## VICTORIA UNIVERSITY OF WELLINGTON Te Whare Wānanga o te Ūpoko o te Ika a Māui



## School of Engineering and Computer Science Te Kura Mātai Pūkaha, Pūrorohiko

PO Box 600 Wellington New Zealand

Tel: +64 4 463 5341 Fax: +64 4 463 5045 Internet: office@ecs.vuw.ac.nz

## **Self Tuning Buck Converter**

Niels Clayton

Supervisors: Daniel Burmester and Ramesh Rayudu

Submitted in partial fulfilment of the requirements for Bachelor of Engineering with Honours.

#### **Abstract**

Switch-mode power supplies are commonly used in a wide variety of consumer and professional appliances to transform DC voltages with high efficiency. One such switch-mode supply is the buck converter, which steps down a DC voltage. The current buck converter design process requires that a specific output filter be designed around the switching frequency of the converter, the required output voltage, and the desired inductor ripple. This filter design process often results in the selection of non-standards components that are difficult to purchase or manufacture, increasing costs and leading to design compromises. This project will implement a control system to actively control the inductor current ripple by modulating the switching frequency of the converter. This will allow engineers to design buck converters directly for the inductor current ripple they wish to tolerate, eliminating the issues of designing this output filter.

# **Contents**

| 1 | Intr | Introduction                                     |    |  |  |  |
|---|------|--------------------------------------------------|----|--|--|--|
|   | 1.1  | Project Motivation                               | 1  |  |  |  |
|   | 1.2  | Project Goals                                    | 1  |  |  |  |
| 2 | Bac  | kground                                          | 3  |  |  |  |
|   | 2.1  | Pulse Width Modulated Signal Generation          | 3  |  |  |  |
|   |      | 2.1.1 Analogue PWM Signal Generation             | 3  |  |  |  |
|   |      | 2.1.2 Digital PWM Signal Generation              | 4  |  |  |  |
|   | 2.2  | Buck Converters                                  | 4  |  |  |  |
|   |      | 2.2.1 Buck Converter Design                      | 5  |  |  |  |
|   | 2.3  | Current Sensing                                  | 6  |  |  |  |
|   |      | 2.3.1 Hall Effect Sensors                        | 6  |  |  |  |
|   |      | 2.3.2 Current Sense Amplification                | 6  |  |  |  |
|   | 2.4  | Control Systems                                  | 6  |  |  |  |
|   |      | 2.4.1 PID Controllers                            | 6  |  |  |  |
| 3 | Des  | ign                                              | 7  |  |  |  |
|   | 3.1  | Defining & Justifying System Specifications      | 7  |  |  |  |
|   | 3.2  | System Architecture & Design                     | 8  |  |  |  |
|   | 3.3  | PWM Generation                                   | 9  |  |  |  |
|   |      | 3.3.1 Analogue PWM Generator Design              | 9  |  |  |  |
|   |      |                                                  | 10 |  |  |  |
|   | 3.4  | Inductor Current Peak to Peak Sensing            | 10 |  |  |  |
|   |      | 3.4.1 Current Sensor Selection                   | 10 |  |  |  |
|   |      | 3.4.2 Precision Rectifier Peak Voltage Detection | 11 |  |  |  |
|   |      | 3.4.3 Sample and Hold Peak Voltage Detection     | 11 |  |  |  |
|   |      | 3.4.4 Current sensing digitisation               | 11 |  |  |  |
|   | 3.5  | J                                                | 11 |  |  |  |
|   |      | 3.5.1 Output Load Voltage Controller             | 11 |  |  |  |
|   |      | 3.5.2 Inductor Current Ripple Controller         | 11 |  |  |  |
| 4 | Imp  | lementation                                      | 12 |  |  |  |
|   | 4.1  |                                                  | 12 |  |  |  |
|   | 4.2  | Inductor Current Peak to Peak Sensor             | 12 |  |  |  |
|   | 4.3  |                                                  | 12 |  |  |  |
|   | 4.4  | ·                                                | 12 |  |  |  |

| 5 | Evaluation                         |    |  |  |
|---|------------------------------------|----|--|--|
|   | 5.1 PWM Generation                 | 13 |  |  |
|   | 5.2 Inductor Current Ripple Sensor | 13 |  |  |
|   | 5.3 Control System                 | 13 |  |  |
| 6 | Conclusions & Future Work          | 14 |  |  |
| A | System Specification Derivation    | 17 |  |  |
| В | PWM Generation Figures             | 19 |  |  |
| C | Peak Detector Figures & Tables     | 22 |  |  |
| D | Control System Figures             | 29 |  |  |
| E | Schematic & Printed Circuit Board  | 34 |  |  |
| F | Source Code                        | 36 |  |  |

# **Chapter 1: Introduction**

Historically power distribution has been primarily in the form of AC (Alternating current). This is credited to the fact that AC power is more efficient to transmit over long distances and made it easy to step up and down the voltages efficiently with transformers [1]. However, with the invention of solid-state electronics such as the metal–oxide–semiconductor field-effect transistor or MOSFET for short, it has become possible to efficiently step up and step down direct current (DC) voltages. This has been achieved by the invention of the switch-mode power supply, which has facilitated the continued reduction of size and increase in efficiency of electronics [2].

Today, switch-mode power supplies can be found in a wide variety of consumer and professional electronics, with some examples being laptops, phones, and any form of DC charger. Their widespread usage when compared to other DC-DC converters such as linear regulators can be attributed to their far greater efficiency. One such switch-mode power supply is the buck converter, which will step down a DC input voltage to a lower DC output voltage.

### 1.1 Project Motivation

Although buck converters are a widespread technology, they are not without their limitations and drawbacks. The current buck converter design process requires that a specific output filter be designed around the switching frequency of the converter, and the desired inductor ripple. This filter design process will often result in the converter requiring discrete passive components that are non-standard and hard to source. This will usually result in the designer having to make compromises in their design for either the cost or the performance of the converter.

Another drawback of this design process is the static nature of both the filter and the switching components once they have been selected. This results in the buck converters desired inductor current ripple only being achieved at a very specific designed output voltage or load. This means that with current buck converter designs varying the desired output voltage or varying the output load will cause the inductor current ripple to vary. This is an issue, as very few loads are static and will not change during their operation.

## 1.2 Project Goals

This project aims to eliminate the need to design the output stage of a buck converter. By implementing a control system that varies the switching frequency of the converter, we will be able to directly manipulate the inductor current ripple. This project aims to produce a proof of concept buck converter that is capable of operating at 12V, with an output range of 3-10V and precision of  $\pm 5\%$ . The converter will also be able vary its switching frequency between 1kHz and 100kHz, allowing for selection of inductor current ripple between 20% and 50% with precision of  $\pm 5\%$ . All of this must be implemented while maintaining the standard functionality of the converter.

TODO Expand on the project goals, possibly convert the previous section into a bulleted list of project goals and requirements.

This will be referenced in the implementation and design sections many times, so it needs to clearly outline what we are looking to achieve in this project.

# **Chapter 2: Background**

A literature research was performed to inform design decision made in this project, and to evaluate any existing research. It will discuss buck converter design factors and topologies, as well as the various different methods of PWM generation. In performing this literature research, we searched Google Scholar, Engineering Village, and Te Waharoa to find designs that utilised variable frequency PWM.

These searches returned no research relevant to the designs of this project, with the only related work focusing on the electromagnetic noise reduction using randomised frequency modulation [3, 4]. Because of this, research was instead performed to inform the design of the buck converter and the generation of PWM signals.

### 2.1 Pulse Width Modulated Signal Generation

Pulse width modulation (PWM) is a digital signal generation technique shown in Figure 2.1a, in which the Period T of the signal is held constant, while the ratio of its logic high period  $T_{on}$  to logic low period  $T_{off}$  is modulated. This ratio of high period to the low period is referred to as the duty cycle of the PWM signal and is often expressed as a percentage, this can be seen in Figure 2.1b.

PWM signals are used in a wide variety of applications for both digital and analogue electronics. PWM is often used to generate analogue signals from digital components by varying the average voltage of the digital PWM signal over time [5]. PWM is also used to control the switching elements contained within switch mode power supplies using this same principle, as discussed in Section 2.2. With regard to this project, we will be looking to generate a PWM signal that can be modulated in both duty cycle and frequency.



Figure 2.1: Pulse width modulated signal characteristics

#### 2.1.1 Analogue PWM Signal Generation

Designing a PWM signal generator using analogue components has three distinct stages required to generate the signal. These stages can be seen in Figure 2.2, and include clock

generation, triangle wave generation, and signal comparator stages [6].

The clock generation stage generates a square wave clock signal at a set frequency. This is usually done using a quartz crystal oscillator, or another form of resonating oscillator circuit. The triangle wave generating state must take the clock signal from the previous stage, and produce a triangle wave of the same frequency. This stage is most often done using a standard op-amp integrating circuit with unity gain at the resonating frequency of the clock source. The final signal comparator stage will convert this triangle wave into a PWM signal. Using a comparator, a reference voltage can be applied to the non-inverting input, and then the triangle wave can be applied to the inverting input. This will produce a pulse train with the same frequency as the clock source, where the period of  $T_{on}$  and  $T_{off}$  is set by the reference voltage.



Figure 2.2: Stages of analogue PWM generation

### 2.1.2 Digital PWM Signal Generation

Designing a PWM signal generator with digital components is far simpler than the method described in Section 2.1.1, and can be done using either a microcontroller or a Field Programmable Gate Array (FPGA). By using an internal timer that is continually incrementing at a known period we can set a period for our PWM. Then by toggling a digital I/O when a compare variable is equal to the value of the timer. we are able to generate a PWM signal with a variable duty cycle [7]. This can be achieved on most microcontrollers, however the maximum frequency and duty cycle accuracy will be dependant on individual clock speed and internal register sizes.

#### 2.2 Buck Converters

The buck converters is a variant of a switch mode power supply that steps down a DC input voltage to a DC output voltage. They are commonly used in a wide variety of consumer and professional appliances such as laptops, phones, and chargers due to their high efficiency compared to other DC-to-DC step down converters such as linear regulators [8].

The basic operational components of a buck converter can be seen below in Figure 2.3. From this we see that a buck converter has three main elements, the input voltage source, two switching components, and an output filter across the load. In the case of Figure 2.3, the first switching component is an actively controlled switch such as a MOSFET or transistor, and the second a passive switching diode. This configuration of an active and a passive switch is known as the non-synchronous buck converter topology, if the passive diode were to be replaced with a second active switch the topology would be considered synchronous. Although both topologies function under the same fundamental principles, the non-synchronous topology is easier to implement with the drawback of higher losses and

therefore lower efficiency.

It can also be seen from Figure 2.3 that a buck converter has two operating states that are controlled through the activation of these switching components. By toggling these switching components at high speed though the use of PWM, we can control the current flowing through the inductor of the output filter. By controlling this current we are also able to directly control the current through, and voltage across the output load of the converter. Using this, buck converters will often have a feedback control system in their design to be able to actively control and regulate the output voltage during usage. This controller will vary the duty cycle of the the switching PWM signal, thereby varying the output voltage of the buck converter as shown in Equation (2.1).



Figure 2.3: Operating states of a buck converter

#### 2.2.1 Buck Converter Design

The design of a common buck converter has two primary considerations, the output voltage of the converter  $V_0$ , and the inductor current ripple of the converter  $\Delta i_L$ . These considerations can be specified by designing the buck converter using Equation (2.1) & Equation (2.2) [9, 10].

When designing a buck converter the first design specification that must be met is the output voltage. In Equation (2.1) the output voltage can be directly related to the input voltage  $V_{in}$  and the switching duty cycle D. Using this equation it is possible to directly set the output voltage of the buck converter by varying this duty cycle.

$$V_o = D \cdot V_{in} \tag{2.1}$$

Once the output voltage has been specified, the inductor current ripple can be calculated and specified with Equation (2.2). This equation allows for the inductor current ripple to be directly related to the inductor size L, and the PWM switching frequency  $f_s$ . This allows the the specification of the inductor current ripple through the varying of these two values.

$$\Delta i_L = \frac{V_o \cdot (1 - D)}{L \cdot f_s} \tag{2.2}$$

These two equations will be used to inform the designs and specifications of this project, and will be discussed in detail in Section 3.1.

### 2.3 Current Sensing

Discuss the varying methods of current sensing that are available, and what each of their advantages and disadvantages are.

#### 2.3.1 Hall Effect Sensors

brief overview of hall effect sensors. They function by measuring the magnetic flux density, meaning that they are commonly used to senses the presence of a magnet. However since a current flowing though a wire will produce a magnetic flux, they can also be used to measure current flow.

Because of this operation, they are able to sense a current without contacting or altering the circuit. This means that they are commonly used in the measurement of high voltage or high current circuits, as they will remain completely isolated from the circuit being sensed, and will also not dissipate power from the sensed circuit. This provides large safety advantages, and can often decrease the complexity of the sensing circuit.

Hall affect sensors do however suffer from a lack of precision. Due to their operation, their measurements will constantly be offset by any ambient magnetic flux. This means that they are not commonly used for the sensing of small signal currents, as the noise floor of the Earth's own magnetic field can often be larger than the signal being sensed.

### 2.3.2 Current Sense Amplification

Current sense amplification works on the basic principle of Ohm's law  $I = \frac{V}{R}$ . By sensing the voltage dropped across a known value resistive element (Often called the current shunt), it is possible to calculate the current that is flowing through the element. This form of current sensing is very simple to implement in theory, however it has the effect of altering the system being sensed by adding a resistive load, and thereby increasing the losses of the system. To mitigate the effects of this sensing on a circuit, a smaller resistive load can be used. However, this will also decrease the measurable voltage across the load, and therefore decrease the precision of a taken measurement. Because of this, shunt based current sensors are often paired with an operational amplifier known gain. This combination allows for accurate amplification of the shunt voltage drop, increasing the precision of the measurement, and allowing for drastically smaller shunt resistors.

## 2.4 Control Systems

Discuss the basics of control theory, and how a controller can be implemented on a digital system within discrete time.

- Discuss in very general terms what a control system is what what it seeks to do in a system.
- Discuss what the control system will be doing in the case of this project. Talk about how a controller will be used to control both the output voltage of the converter, and the inductor ripple of the converter.
- Discuss the design, use, and implementation of a PID controller within a system.

#### 2.4.1 PID Controllers

# Chapter 3: Design

The deign processes outlined within this report will draw heavily on the background research discussed in Chapter 2. These designs aim to effetely implement the outlined system requirements in a robust and repeatable manner.

## 3.1 Defining & Justifying System Specifications

Based on the system requirements that have been outlined in Chapter 1, a set of system specifications can be created to inform our design decisions.

It has been specified that the final system design will be able to select for both the output load voltage, and the inductor current ripple of the buck converter. From this requirement we can identity that two separate control systems should be designed, one to regulate the output voltage of the converter, and one to regulate the inductor current ripple.

Next we can identify that to control the output load voltage and the inductor current ripple, we must be able to actively effect their current state. Based on Equation (2.1) we can see that by varying the duty cycle of the converters PWM signal, we are able to directly control the output voltage. Similarly, from Equation (2.2) we can see that by varying the converters PWM switching frequency we are able to directly control the inductor current ripple. From this we can specify that our designs of the PWM generation must be capable of varying both the duty cycle and the switching frequency of the output PWM signal independently and simultaneously.

The outlined requirements also specify the level of precision that will be required from the output load voltage and the inductor current ripple. This allows us to specify the tolerable error for the system design, and calculate the minimum required system specifications. Based on the buck converter design equations from Section 2.2.1, The following specifications have been identified:

• Maximum allowable duty cycle step size

$$V_{error} = V_{min} \cdot error = 0.15V \tag{3.1}$$

$$D_{step} = \frac{V_{error}}{V_{in}} = 0.0125 \tag{3.2}$$

$$N_{step} = \frac{1}{D_{step}} = 80 \tag{3.3}$$

• Maximum & minimum inductor sizes

$$L_{max} = \frac{V_{max} \cdot (1 - D_{max})}{f_{min} \cdot I_{min}} = 27.7mH$$
 (3.4)

$$L_{min} = \frac{\frac{V_{in}}{2} \cdot (1 - 0.5)}{f_{max} \cdot I_{min}} = 0.5mH$$
 (3.5)

• Maximum allowable frequency step size

$$f_{step} = \frac{V_{max} \cdot (1 - D_{max})}{(I_{min} - I_{Error}) \cdot L_{max}} - f_{min} = 52Hz$$
(3.6)

$$N_{steps} = \frac{(f_{max} - f_{min})}{f_{step}} = 1881 \tag{3.7}$$

The derivation of these values can be found in Appendix A.

#### TODO Refactor this to be a list of specifications

From these equations we can build a list of final specifications to inform the design of our PWM generator and buck converter. The PWM generator must provide a minimum voltage step size of 0.0125V, for a resolution of 80 voltage steps between 3V and 10V. The PWM generator must also be able to provide a minimum frequency step size of 52Hz, for a resolution of 1881 frequency steps between 1kHz & 100kHz. Finally we can also specify that the buck converter must be capable of functioning with inductor values between 0.5mH & 27.7mH.

By designing the PWM generator and the buck converter to these specifications, we are able to guarantee that we can always achieve the requirements outlined in Chapter 1.

# TODO Discuss the minimum and maximum sense current, along with the bandwidth requirement of the sensors

Design the system specifications around the current sensing and voltage sensing. What is the required bandwidth of the sensor? What is the required precision?

## 3.2 System Architecture & Design

To achieve the specifications that have been outlined in Section 3.1, it is important to design the system architecture around them. In Figure 3.1 an overview of the system architecture can be seen, with three main design sections outlined. These sections each represent a significant segment of work that must be completed for the final artefact of this project to be achieved.

The first section of work that must be completed is the design of the PWM generation, denoted 1 in Figure 3.1. This PWM generator will be used to control both the output voltage and the inductor current ripple, and as such must be able to modulate both the duty cycle and the frequency of the PWM to the precisions required.

The second section of work is the design of the sensing elements required by the system, denoted 2 in Figure 3.1. These elements will be used to measure both the output voltage and the inductor current ripple, and therefore must be able to achieve the required precisions and sampling rates.

Finally the third section of work is the design and implementation of the two control systems, denoted 3 in Figure 3.1. These control systems will be responsible for maintaining the desired output voltage and inductor current ripple of the buck converter. This system will therefore be responsible for facilitating the final functionality of the project, combining sections 1 & 2.



Figure 3.1: High level system overview

### 3.3 PWM Generation

In the design of the PWM generator, both the analogue and digital designs discussed in Section 2.1 were considered, designed, and tested for. Each of these designs presented pros and cons that would affects the overall design of the system architecture. This section will discuss these designs and finalise the design of the PWM generator for this system.

#### 3.3.1 Analogue PWM Generator Design

As discussed in Section 2.1.1, the design of the analogue PWM signal generator requires three stages, each of which will have its own design requirements based on the specifications outlined in Section 3.1.

The clock generation stage will be responsible for setting the frequency of the final PWM signal. This specifies that the clock source have a variable frequency output range between 1kHz and 100kHz, with a minimum step size of 52Hz. Research was done on a variety of clock sources, looking at voltage-controlled oscillators (VCO's), signal generator IC's, and even the basic 555 timer. From this VCO's were identified to operate at much higher frequencies than those used in this project. It was also identified that signal generator IC's

often require selections of passive components to operate effectively, increasing their complexity. For this reason the variable frequency 555 timer circuit was selected, as it provided the required specifications.

The next section designed was the signal integrator stage. This stage consisted of a basic op-amp integrator circuit, with a design requirement that it be able to integrate the clock signal across the frequency range required. This circuit was designed and implemented in an LTSpice simulation to evaluate it's performance, and can be seen in Appendix B. From this simulation it was noted that the integrator's frequency response was similar to that of a first order low pass filter, and greatly attenuated the integrated signal. For this reason it was decided that analogue PWM generation would not be implemented in this system, as it presented many issues.

# TODO insert images of the designed integrator circuit, and it's frequency response (bode plot)

#### 3.3.2 Digital PWM Generator Design

As discussed in Section 2.1.2, The design of the digital PWM generator is far simpler than that of the analogue, and can be implemented in a wide variety of methods. In this project microcontrollers and FPGA's have been considered.

Based solely on the capabilities of the platform, the PWM generator would be best designed and implemented on an FPGA as it would allow for superior speed and precision. However FGPA design brings a lot of difficulties, primarily in the prototyping and testing stages. Because of this, due to the limited time from of this project, we have decided to implement this PWM design using a microcontroller.

The selection of the microcontroller is highly dependant on the clock frequency and design of the PWM peripherals, as it must be capable of achieving the specifications outlined in Section 3.1. A large selection of microcontroller data-sheets were reviewed to identify their specifications, including AVR, STM8, Espressif, and teensy based microcontrollers. From this review it was decided that the ESP32 microcontroller would be best suited to this project [11]. This microcontroller is capable of outputting a maximum PWM frequency 125kHz with a duty cycle resolution of 9 bits (512 voltage steps). From here a short C program was written to test the PWM functionality of the ESP32, and it was confirmed that it met the required specifications. The source code and images of this PWM signal can be found in Appendix B.

## 3.4 Inductor Current Peak to Peak Sensing

Discuss the purpose of this sensors, and the requirements of the sensing system in regards to its range, precision, and bandwidth. Also discuss the required outputs of the sensor (we need to detects the current peak value, and the mean current).

#### 3.4.1 Current Sensor Selection

Talk about how hall effect sensors were identified to not be suitable for this design due to the reasons outlined in Section 2.3.1. From here the discuss the selection of a current sense amplifier (it's gain, bandwidth, and precision), and then the selection and design of the shunt resistor. Then discuss the expected voltage output of this current and the expected waveform

Based on the specifications outlined in Section 3.1, it can be identified that directly sampling the output waveform from this sensors is not feasible due to it's large possible bandwidth.

From there then discuss the design of varying analog circuits to attempt to identify the peak current and mean current.

I want to show circuit designs, and simulations for both the precision rectifier peak detection design, and the capture and hold peak detection design.

#### 3.4.2 Precision Rectifier Peak Voltage Detection

Design, and simulations of the Precision Rectifier Peak Voltage Detection.

#### 3.4.3 Sample and Hold Peak Voltage Detection

Design, and simulations of the Sample and Hold Peak Voltage Detection.

### 3.4.4 Current sensing digitisation

Discuss the selection of the ADC for measuring the peak ripple and mean ripple currents. Talk about the minimum required ADC precision, and bandwidth requirements.

### 3.5 Control System

The control system designs will inform the selection of the sensors used within the system design. This section will cover the selection of the controller topology (PI & PID), some basic modelling of the system, and then the selection of the sensors required to generate the correct feedback signals.

#### 3.5.1 Output Load Voltage Controller

#### 3.5.2 Inductor Current Ripple Controller

# **Chapter 4: Implementation**

#### 4.1 PWM Generation

go ask Daniel about this because currently all I can think to say is about the bootstrap circuit and the code. But the bootstrap circuit is not important to the design, and was inly used due to covid since I will be using a gate driver in the final design.

#### **TODO Gate driving and Bootstrapping**

Discuss how the micro will be unable to directly drive the gate of the switching MOSFET, as it's  $V_{GS_{on}}$  will be too large. Because of this a gate driver will need to be selected. The driver will need to be driven using the 3.3V logic from the ESP32, and will need timings fast enough to function correctly at 100kHz. I can also discuss how during the lock-down I was unable to purchase a gate driver, and so I built a bootstrapping circuit from components I located around the house.

#### 4.2 Inductor Current Peak to Peak Sensor

Do I just show a photo of the breadboard and say that I built it on that?

## 4.3 Control System

#### 4.4 Software Architecture

Discuss the ESPidf HAL (Hardware Abstraction Layer), and how it provides greater control of the system resources than the more commonly used arduino platform.

Then discuss how the system operates within the freeRTOS real time operating system, what allows for easy multitasking between the different time sensitive control loops that will have to be run on the micro.

Finally discuss the simple to use API that was implemented, which aims to abstract away the HAL layer. This API is implemented using collection of single header libraries written in the 'C' programming language, with each implementing only the core functionality of the hardware they are interacting with. This makes the software platform robust and simple to move to different embedded platforms, with such a move only requiring the re-implementation of the core functions of each library.

# **Chapter 5: Evaluation**

The evaluation of the system will be conducted using a range of load resistances, evaluating its performance with  $10\Omega$ ,  $15\Omega$ , and  $20\Omega$  output loads, using a constant supply voltage of 12V DC. S

- The buck converter will be able to take input voltages up to 12V DC
- The buck converter must maintain the basic functionality outlined in eq. (2.1)
- The buck converter will have an output voltage range between 3V and 10V DC
- The output voltage accuracy will be within  $\pm 5\%$  of the target output voltage
- The user will be able to define the inductor ripple between 20% and 50%
- The inductor ripple accuracy will be within  $\pm 5\%$  of the defined inductor ripple
- The buck converter will have a switching frequency range of 1kHz to 100kHz
- The control system will have no steady state error
- 5.1 PWM Generation
- 5.2 Inductor Current Ripple Sensor
- 5.3 Control System

# **Chapter 6: Conclusions & Future Work**

# **Bibliography**

- [1] E. Earley, "What's the difference between ac and dc?." Online, Sept. 2013.
- [2] G. Bocock, "History of switch mode power supplies (smps)." Online.
- [3] J. R. Roman, "PWM regulator with varying operating frequency for reduced EMI," Mar. 2001.
- [4] Y. L. Familiant and A. Ruderman, "A Variable Switching Frequency PWM Technique for Induction Motor Drive to Spread Acoustic Noise Spectrum With Reduced Current Ripple," *IEEE transactions on industry applications*, vol. 52, no. 6, pp. 5355–5355, 2016.
- [5] W. Tareen, M. Aamir, S. Mekhilef, M. Nakaoka, M. Seyedmahmoudian, B. Horan, M. A. Memon, and N. A. Baig, "Mitigation of power quality issues due to highpenetration of renewable energy sources in electricgrid systems using three-phase apf/statcomtechnologies: A review," in *Power Electronics in Renewable Energy Systems* (T. Suntio and T. Messo, eds.), pp. 105–133, 2019.
- [6] J. Caldwell, Analog Pulse Width Modulation. texas instruments, June 2013.
- [7] S. Colley, "Pulse-width modulation (pwm) timers in microcontrollers." Online, Feb. 2020.
- [8] N. Mohan, Power Electronics: A First Course. Don Fowley, Oct. 2011.
- [9] N. Mohan, *Power Electronics a First Course*, ch. 4: Switch Mode DC-DC Converters: Switching Analysis, Topology Selection and Design, pp. 38–68. Don Fowley, 2012.
- [10] B. Hauke, "Basic Calculation of a Buck Converter's Power Stage," tech. rep., Texas Instruments, Aug. 2015.
- [11] Espressif Systems, ESP32: Technical Reference Manual, 4.4 ed., 2021.

# Acknowledgments

Thank you Danny B for being great :- )
Thank you ECS Techs for putting up with my questions
And thank you to my wonderful girlfriend for not kicking me out when I was working on this stupid fucking project really late

# Appendix A: System Specification Derivation

### **PWM Duty Cycle Resolution Calculations**

It has been specified that the output voltage error should be within  $\pm 5\%$  of the output voltage. This allowable error will be smallest with the smallest output voltage, and will therefore define the smallest allowable duty cycle resolution.

Calculate the allowable output voltage error:

$$V_{error} = 3 \cdot 0.05$$

$$= 0.15V$$
(A.1)

Calculate the minimum number of discrete voltage steps based on Equation (2.1)

$$N_{steps} = \frac{V_{in}}{V_{error}} = \frac{12}{0.15}$$

$$= 80$$
(A.2)

Calculate the resolution in bits:

$$N_{bits} = \log_2(N_{steps}) = \log_2(80)$$

$$= 6.3 \approx 7$$
(A.3)

This gives a minimum duty cycle resolution of 7 bits, or 128 discrete steps.

#### Minimum & Maximum Inductor Size Calculations

It has been specified that the operating frequency of the buck converter should be between 1kHz & 100kHz, with an inductor current ripple between 20% & 50%. Based on these specifications, Equation (2.2) can be used to derive the minimum and maximum inductor sizes that the system will continue to function with.

#### **Derivation Constants:**

$$R_{load}=10\Omega,\,V_{in}=12V,\,V_{max}=10V,\,V_{min}=3V,\,f_{min}=1\text{kHz},\,f_{max}=100\text{kHz}$$

All calculations will be performed using the minimum allowable inductor current ripple of 20%, calculated below. This will ensure that all greater current ripple percentages are achievable by the system.

$$i_{min} = \frac{V_{min}}{R_{load}}$$

$$= 0.3A$$
(A.4)

$$\Delta i_{min} = i_{min} \cdot 0.2 \tag{A.5}$$

#### **Minimum Inductor Size**

Using Equation (A.6), we see that the minimum achievable indictor size will be at the maximum switching frequency. It can be noted that due to the quadratic nature of this equation, the maximum switching frequency for a constant ripple will occur at an output of  $V_0 = \frac{v_{in}}{2}$ , with a duty cycle of 50%.

= 0.06A

$$L = \frac{V_o \cdot (1 - D)}{\Delta i_L \cdot f_s} \tag{A.6}$$

Calculate the minimum inductor value:

$$L_{min} = \frac{\frac{v_{in}}{2} \cdot (1 - 0.5)}{\Delta i_{min} \cdot f_{max}}$$

$$= \frac{6 \cdot (1 - 0.5)}{100000 \cdot 0.06}$$

$$= 0.5mH$$
(A.7)

#### **Maximum Inductor Size**

From Equation (A.6), we can see that the maximum inductor value will be at the lowest switching frequency, with the greatest output voltage, and therefore the greatest duty cycle.

Calculate the maximum duty cycle:

$$D_{max} = \frac{V_{max}}{V_{in}}$$

$$= \frac{10}{12} = 83.3\%$$
(A.8)

Calculate the maximum inductor size:

$$L_{min} = \frac{V_{max} \cdot (1 - D_{max})}{\Delta i_{min} \cdot f_{min}}$$

$$= \frac{1 \cdot (1 - 0.83)}{1000 \cdot 0.06}$$

$$= 27.78mH$$
(A.9)

# **Appendix B: PWM Generation Figures**

## **Analogue PWM Generation**



Figure B.1: Operational amplifier integrator LTSpice circuit simulation



Figure B.2: Operational amplifier integrator simulation with a 1kHz square wave input. Output triangle wave amplitude of 1.5V



Figure B.3: Operational amplifier integrator simulation with a 1kHz square wave input. Output triangle wave amplitude of 20mV

## **Digital PWM Generation**

## **PWM Duty Cycle Control**



(a) Digital PWM Generation with a 20% duty (b) Digital PWM Generation with a 50% duty cycle  $\,$  cycle



(c) Digital PWM Generation with a 90% duty cycle

Figure B.4: Digital PWM: Varying PWM duty cycle with a 100kHz frequency

### **PWM Frequency Control**



(a) Digital PWM Generation with a 1kHz fre- (b) Digital PWM Generation with a 50kHz quency



(c) Digital PWM Generation with a 100kHz frequency

Figure B.5: Digital PWM: Varying PWM frequency with a 50% duty cycle

# **Appendix C:** Peak Detector Figures & Tables

## Initial & Final Peak Detector Output Comparison

### Peak Detector Output for a 150mV Peak to Peak Triangle Input



(c) Initial (Red) & final (Blue) peak detetor (d) Initial (Red) & final (Blue) peak detetor outputs for a 50kHz triangle input. outputs for a 100kHz triangle input.

Figure C.1: Initial & Final Peak Detector Output at varying frequencies for a 150mV peak to peak triangle input.

### Peak Detector Output for a 500mV Peak to Peak Triangle Input



(a) Initial (Red) & final (Blue) peak detetor (b) Initial (Red) & final (Blue) peak detetor outputs for a 1kHz triangle input. outputs for a 10kHz triangle input.



(c) Initial (Red) & final (Blue) peak detetor (d) Initial (Red) & final (Blue) peak detetor outputs for a 50kHz triangle input. outputs for a 100kHz triangle input.

Figure C.2: Initial & Final Peak Detector Output at varying frequencies for a 500mV peak to peak triangle input.

### Peak Detector Output for a 1500mV Peak to Peak Triangle Input



(c) Initial (Red) & final (Blue) peak detetor (d) Initial (Red) & final (Blue) peak detetor outputs for a 50kHz triangle input. outputs for a 100kHz triangle input.

Figure C.3: Initial & Final Peak Detector Output at varying frequencies for a 1500mV peak to peak triangle input.

### **Initial & Final Peak Detector Output Error Plots**



Figure C.4: Initial & final peak detector output error across frequencies for a 150mV peak to peak input. Displayed as a percentage of the ideal output.



Figure C.5: Initial & final peak detector output error across frequencies for a 500mV peak to peak input. Displayed as a percentage of the ideal output.



Figure C.6: Initial & final peak detector output error across frequencies for a 1500mV peak to peak input. Displayed as a percentage of the ideal output.

## **Initial & Final Peak Detector Output Error Tables**

| 150mV Peak to Peak Input Signal<br>Initial and Final Design Peak Detector Output Error Across Frequencies             |       |              |       |             |  |
|-----------------------------------------------------------------------------------------------------------------------|-------|--------------|-------|-------------|--|
| Frequency (kHz)   Final Design Error (mV)   Final Design % Error   Initial Design Error (mV)   Initial Design % Error |       |              |       |             |  |
| 1                                                                                                                     | -3.90 | -5.2         | 3.40  | 4.533333333 |  |
| 5                                                                                                                     | -3.80 | -5.066666667 | 5.80  | 7.733333333 |  |
| 10                                                                                                                    | -3.70 | -4.933333333 | 12.80 | 17.06666667 |  |
| 15                                                                                                                    | -2.90 | -3.866666667 | 18.50 | 24.66666667 |  |
| 20                                                                                                                    | -2.20 | -2.933333333 | 23.30 | 31.06666667 |  |
| 25                                                                                                                    | -1.60 | -2.133333333 | 28.30 | 37.73333333 |  |
| 30                                                                                                                    | -0.90 | -1.2         | 32.60 | 43.46666667 |  |
| 35                                                                                                                    | -0.30 | -0.4         | 36.50 | 48.66666667 |  |
| 40                                                                                                                    | 0.20  | 0.266666667  | 40.20 | 53.6        |  |
| 45                                                                                                                    | 0.80  | 1.066666667  | 43.70 | 58.26666667 |  |
| 50                                                                                                                    | 1.30  | 1.733333333  | 47.00 | 62.66666667 |  |
| 55                                                                                                                    | 2.00  | 2.666666667  | 50.10 | 66.8        |  |
| 60                                                                                                                    | 2.60  | 3.466666667  | 53.00 | 70.66666667 |  |
| 65                                                                                                                    | 3.10  | 4.133333333  | 56.10 | 74.8        |  |
| 70                                                                                                                    | 3.50  | 4.666666667  | 59.50 | 79.33333333 |  |
| 75                                                                                                                    | 4.00  | 5.333333333  | 62.70 | 83.6        |  |
| 80                                                                                                                    | 4.60  | 6.133333333  | 65.20 | 86.93333333 |  |
| 85                                                                                                                    | 5.00  | 6.666666667  | 67.80 | 90.4        |  |
| 90                                                                                                                    | 5.60  | 7.466666667  | 68.10 | 90.8        |  |
| 95                                                                                                                    | 6.10  | 8.133333333  | 68.10 | 90.8        |  |
| 100                                                                                                                   | 6.70  | 8.933333333  | 68.10 | 90.8        |  |

Table C.1: Table of output errors at varying frequencies for both the initial and final peak detection design with a 150mV peak to peak input signal.

| 500mV Peak to Peak Input Signal Initial and Final Design Peak Detector Output Error Across Frequencies |       |       |                           |                        |
|--------------------------------------------------------------------------------------------------------|-------|-------|---------------------------|------------------------|
| Frequency (kHz)                                                                                        |       |       | Initial Design Error (mV) | Initial Design % Error |
| 1                                                                                                      | -1.60 | -0.64 | 7.00                      | 2.8                    |
| 5                                                                                                      | -1.70 | -0.68 | 15.00                     | 6                      |
| 10                                                                                                     | -1.90 | -0.76 | 23.90                     | 9.56                   |
| 15                                                                                                     | -0.50 | -0.2  | 32.90                     | 13.16                  |
| 20                                                                                                     | 1.40  | 0.56  | 40.70                     | 16.28                  |
| 25                                                                                                     | 2.70  | 1.08  | 47.90                     | 19.16                  |
| 30                                                                                                     | 3.30  | 1.32  | 54.40                     | 21.76                  |
| 35                                                                                                     | 4.40  | 1.76  | 60.60                     | 24.24                  |
| 40                                                                                                     | 5.50  | 2.2   | 66.30                     | 26.52                  |
| 45                                                                                                     | 6.30  | 2.52  | 71.90                     | 28.76                  |
| 50                                                                                                     | 7.40  | 2.96  | 77.30                     | 30.92                  |
| 55                                                                                                     | 8.30  | 3.32  | 82.40                     | 32.96                  |
| 60                                                                                                     | 9.40  | 3.76  | 87.40                     | 34.96                  |
| 65                                                                                                     | 10.20 | 4.08  | 92.50                     | 37                     |
| 70                                                                                                     | 11.00 | 4.4   | 97.00                     | 38.8                   |
| 75                                                                                                     | 11.80 | 4.72  | 101.70                    | 40.68                  |
| 80                                                                                                     | 12.70 | 5.08  | 106.20                    | 42.48                  |
| 85                                                                                                     | 13.40 | 5.36  | 110.50                    | 44.2                   |
| 90                                                                                                     | 14.40 | 5.76  | 114.90                    | 45.96                  |
| 95                                                                                                     | 15.00 | 6     | 119.00                    | 47.6                   |
| 100                                                                                                    | 15.80 | 6.32  | 123.20                    | 49.28                  |

Table C.2: Table of output errors at varying frequencies for both the initial and final peak detection design with a 500mV input signal.

| 1500mV Peak to Peak Input Signal<br>Initial and Final Design Peak Detector Output Error Across Frequencies |                         |                      |        |                        |  |
|------------------------------------------------------------------------------------------------------------|-------------------------|----------------------|--------|------------------------|--|
| Frequency (kHz)                                                                                            | Final Design Error (mV) | Final Design % Error |        | Initial Design % Error |  |
| 1                                                                                                          | 5.40                    | 0.72                 | 15.00  | 2                      |  |
| 5                                                                                                          | 5.70                    | 0.76                 | 29.70  | 3.96                   |  |
| 10                                                                                                         | 3.60                    | 0.48                 | 41.70  | 5.56                   |  |
| 15                                                                                                         | 6.50                    | 0.866666667          | 55.90  | 7.453333333            |  |
| 20                                                                                                         | 9.40                    | 1.253333333          | 68.70  | 9.16                   |  |
| 25                                                                                                         | 11.60                   | 1.546666667          | 79.90  | 10.65333333            |  |
| 30                                                                                                         | 13.50                   | 1.8                  | 90.00  | 12                     |  |
| 35                                                                                                         | 15.10                   | 2.013333333          | 99.90  | 13.32                  |  |
| 40                                                                                                         | 17.20                   | 2.293333333          | 108.70 | 14.49333333            |  |
| 45                                                                                                         | 19.20                   | 2.56                 | 117.20 | 15.62666667            |  |
| 50                                                                                                         | 20.50                   | 2.733333333          | 125.50 | 16.73333333            |  |
| 55                                                                                                         | 21.90                   | 2.92                 | 133.60 | 17.81333333            |  |
| 60                                                                                                         | 23.30                   | 3.106666667          | 141.50 | 18.86666667            |  |
| 65                                                                                                         | 24.80                   | 3.306666667          | 149.20 | 19.89333333            |  |
| 70                                                                                                         | 26.10                   | 3.48                 | 158.20 | 21.09333333            |  |
| 75                                                                                                         | 27.50                   | 3.666666667          | 163.80 | 21.84                  |  |
| 80                                                                                                         | 28.60                   | 3.813333333          | 170.70 | 22.76                  |  |
| 85                                                                                                         | 29.60                   | 3.946666667          | 177.50 | 23.66666667            |  |
| 90                                                                                                         | 30.70                   | 4.093333333          | 184.40 | 24.58666667            |  |
| 95                                                                                                         | 32.10                   | 4.28                 | 191.00 | 25.46666667            |  |
| 100                                                                                                        | 33.40                   | 4.453333333          | 197.50 | 26.33333333            |  |

Table C.3: Table of output errors at varying frequencies for both the initial and final peak detection design with a 1500 mV input signal.

# **Appendix D: Control System Figures**

## Variable Output Voltage Duty Cycle Control



(a) Controlled duty cycle of 28.9% for a 3V DC output



(b) Controlled duty cycle of 45.25% for a 5V DC output



(c) Controlled duty cycle of 86.0% for a 10V DC output

Figure D.1: PWM duty cycle controlled by the voltage output control system (Blue), achieving 3V, 5V and 10V DC (Red).

## Output Voltage Control System Step Response Rise Time



(a) Output voltage controller 0V to 1V step response with 86ms settleing time.



(b) Output voltage controller 0V to 3V step response with 84 ms settleing time.



(c) Output voltage controller 0V to 10V step response with 83ms settleing time.

Figure D.2: Output voltage control system step response, reacting to 1V, 3V and 10V output steps.

## Output Voltage Control System Step Response Fall Time



(a) Output voltage controller 3V to 0V step response with  $121 \mathrm{ms}$  settleing time.



(b) Output voltage controller 10V to 0V step response with 113ms settleing time.

Figure D.3: Output voltage control system step response, reacting to -3V and -10V output steps.

## Output Voltage Control System with Variable Supply Voltage



(a) Output voltage controller regulating a 6V supply (Red) for a 3V output (Blue)



(b) Output voltage controller regulating a 12V supply (Red) for a 3V output (Blue)



(c) Output voltage controller regulating a 16V supply (Red) for a 3V output (Blue)

Figure D.4: Output voltage controller regulating variable supply voltages for constant output.

## Output Voltage Control System Supply Voltage Step



Figure D.5: Output voltage controller response (Blue) to a 12V to 8V supply voltage step (Red), with a recovery time of 290ms



Figure D.6: Output voltage controller response (Blue) for a 8V to 12V supply voltage step (Red), with a recovery time of 270ms

# Appendix E: Schematic & Printed Circuit Board

## **Full System Schematic**



Figure E.1: Final system full schematic

## **Designed Printed Circuit Board**



Figure E.2: Final system printed circuit board rendering

# **Appendix F:** Source Code

## Single Header Libraries: Hardware Drivers

#### adc.h

39

```
1 #ifndef BUCK_CONVERTER_ADC
2 #define BUCK_CONVERTER_ADC
4 #include <math.h>
5 #include "driver/adc.h" // Include the ADC driver
7 // ADC struct to hold setup data
8 typedef struct esp_adc
      // Rolling average buffer
10
     uint16_t *buffer;
11
     uint8_t span;
13
     // ADC channel to sample
14
     uint8_t adc_channel;
17 } esp_adc;
19 #endif // BUCK_CONVERTER_ADC
21
22 #ifdef BUCK_CONVERTER_ADC_IMPL
23
* Set up the adc to have a 12bit width, and 11dB attenuation
27 esp_err_t adc_init(esp_adc* adc){
      // ADC set up
29
      esp_err_t status; // Status variable to check if adc initialisation was
30
31
      status = adc1_config_width(ADC_WIDTH_BIT_12); // Set adc width to 12 bits
     if(status != ESP_OK) return status;
      status = adc1_config_channel_atten(adc->adc_channel, ADC_ATTEN_11db); // Set
      adc attenuation to 11dB
      return status;
36
37 }
```

```
40 /*
* Read the raw value from the adc
43 uint16_t adc_read(esp_adc* adc){
      // Read the raw value from the provided adc channel
      return adc1_get_raw(adc->adc_channel);
46 }
47
48 /* Compute the rolling average of the raw ADC values.
  * The circular buffer will store the index of the next element to be
  * replaced in it's final index.
51
52 float rolling_average(esp_adc* adc, int raw_value){
      // Check if there is an averaging buffer
      if(adc->span == 0)
55
          printf("Can not compute average with a span of 0\n");
          return -1;
58
      }
59
      // get the index of the value to replace, and then replace it
      int next_index = adc->buffer[adc->span];
62
      adc->buffer[next_index] = raw_value;
      // Increment the index of the last value and check if it past the end of the
      array
      next_index++;
66
      adc->buffer[adc->span] = (next_index >= adc->span) ? 0 : next_index;
68
      // Calculate the rolling average
      float total = 0;
70
      for(int i = 0; i < adc->span; i++)
          total = total + adc->buffer[i];
73
74
      return total/adc->span;
76
77 }
79 /* Compute the conversion from the adc reading to a voltage.
80
  * The output of the adc has been characterised and then a polynomial has been fit
      to the curve.
  * This provides an error of < 1% for readings between 200mV and 3100mV
  float IRAM_ATTR adc_conversion(float acd_reading)
      return (pow(acd_reading , 4) * -7.6813211494455e-15) +
              (pow(acd\_reading , 3) * 5.03088719249885e-11) +
87
             (pow(acd\_reading , 2) * -1.06609443189713e-7) +
88
             (0.00085850726668 * acd_reading) + 0.09077205072441;
90 }
92 #endif
```

code/adc.h

### pid.h

```
1 #ifndef BUCK_CONVERTER_PID
2 #define BUCK_CONVERTER_PID
4 /*
* PID Controller code structure designed by https://github.com/pms67 under the MIT
      licence
6 * https://github.com/pms67/PID
7 */
8 typedef struct
9 {
10
      /* Controller gains */
11
      float Kp;
12
      float Ki;
13
14
      float Kd;
      /* Derivative low-pass filter time constant */
      float tau;
      /* Output limits */
      float limMin;
20
      float limMax;
21
22
      /* Integrator limits */
      float limMinInt;
24
      float limMaxInt;
      /* Sample time (in seconds) */
      float T;
28
29
      /* Controller "memory" */
     float integrator;
      float prevError; /* Required for integrator */
      float differentiator;
      float prevMeasurement; /* Required for differentiator */
      /* Controller output */
36
      float out;
37
39 } PIDController;
41 #endif
43
44 #ifdef BUCK_CONVERTER_PID_IMPL
46 void PID_controller_init(PIDController *pid){
47
      /* Clear controller variables */
      pid->integrator = 0.0f;
      pid->prevError = 0.0f;
51
      pid->differentiator = 0.0f;
52
      pid->prevMeasurement = 0.0f;
```

```
54
      pid->out = 0.0f;
55
56 }
57
58
59 void IRAM_ATTR PID_controller_update(PIDController *pid, float setpoint, float
      measurement){
60
      // Error Signal
61
62
      float error = setpoint - measurement;
63
      // Proportional Signal
64
      float proportional = pid->Kp * error;
65
      // Integral Signal
67
      pid->integrator = pid->integrator + (pid->Ki * pid->T * (error +
      pid->prevError) * 0.5f);
      /* Anti-wind-up via integrator clamping */
70
      if (pid->integrator > pid->limMaxInt){
71
72
           pid->integrator = pid->limMaxInt;
      }
73
74
      else if (pid->integrator < pid->limMinInt){
           pid->integrator = pid->limMinInt;
77
78
      // Derivative Signal with low pass filtering
79
      pid->differentiator = -(2.0f * pid->Kd * (measurement - pid->prevMeasurement)
      /* Note: derivative on measurement, therefore minus sign in front of equation!
                              +(2.0f * pid->tau - pid->T) * pid->differentiator) /
81
                               (2.0f * pid->tau + pid->T);
82
83
84
     * Compute output and apply limits
85
86
      pid->out = proportional + pid->integrator + pid->differentiator;
87
88
       if (pid->out > pid->limMax){
89
           pid->out = pid->limMax;
91
92
      else if (pid->out < pid->limMin){
93
           pid->out = pid->limMin;
94
95
       /* Store error and measurement for later use */
      pid->prevError = error;
98
      pid->prevMeasurement = measurement;
99
100 }
101
102 #endif //BUCK_CONVERTER_PID
```

code/pid.h

### pwm.h

```
1 #ifndef BUCK_CONVERTER_PWM
2 #define BUCK_CONVERTER_PWM
4 // Hardware driver includes
5 #include "driver/ledc.h"
7 /*
8 * PWM set up defines and helper macros
10 #define DUTY_RESOLUTION LEDC_TIMER_9_BIT
11 #define FREQUENCY_MIN
                          1000
12 #define FREQUENCY_MAX
                          100000
13 #define DUTY_STEPS
                          512
                          ((1.0 - X) * DUTY_STEPS) // Duty cycle calculation macro
14 #define DUTY(X)
15
16 #endif
17
19 #ifdef BUCK_CONVERTER_PWM_IMPL
22 *
       Set up the PWM timer and channel structs, and configure the PWM output
23 */
25 void PWM_setup(ledc_timer_config_t *pwm_timer, ledc_channel_config_t *pwm_channel,
      uint32_t frequency, float duty_cycle){
26
      // Select and set up the PWM clock source and initial frequency
      ledc_timer_config_t timer = {
28
          .duty_resolution = DUTY_RESOLUTION,
                                                  // resolution of PWM duty
29
          .freq_hz = frequency,
                                                  // frequency of PWM signal
          .speed_mode = LEDC_HIGH_SPEED_MODE,
                                                 // timer mode
          .timer_num = LEDC_TIMER_0,
                                                 // timer index
                                                 // Auto select the source clock
          .clk_cfg = LEDC_AUTO_CLK,
33
      };
35
      // Select and set up the PWM output channel and initial duty cycle
36
      ledc_channel_config_t channel = {
37
                                                // PWM output channel (0 - 4)
          .channel = LEDC_CHANNEL_0,
                      = DUTY(duty_cycle),
                                                // PWM duty cycle
          .duty
          .gpio_num = 23,
                                                 // PWM output GPIO
40
          .speed_mode = LEDC_HIGH_SPEED_MODE,
                                                 // PWM output mode
41
          .timer_sel = LEDC_TIMER_0
                                                  // PWM timer index
43
      };
44
45
      *pwm_timer = timer;
      *pwm_channel = channel;
47
      ledc_timer_config(pwm_timer);
48
      ledc_channel_config(pwm_channel);
49
50 }
51
52 /*
       Set a new PWM duty cycle for a given PWM channel
```

```
*/
55
56 esp_err_t PWM_set_duty(ledc_channel_config_t *pwm_channel, float duty_cycle){
57
      // Set the new PWM Duty Cycle of the given channel configuration
58
      ledc_set_duty(pwm_channel -> speed_mode, pwm_channel -> channel,
      DUTY(duty_cycle));
60
      // Update the PWM channel with the new configuration
61
      return ledc_update_duty(pwm_channel -> speed_mode, pwm_channel ->channel);
62
63 }
64
65
66 /*
       Set a new PWM frequency for a given PWM channel
   */
68
69
70 esp_err_t PWM_set_frequency(ledc_channel_config_t *pwm_channel, ledc_timer_config_t
      *pwm_timer, uint32_t frequency){
71
      if(frequency > FREQUENCY_MAX || frequency < FREQUENCY_MIN)</pre>
72
73
          printf("Specified input frequency not within possible range: %d - %d\n",
74
      FREQUENCY_MIN, FREQUENCY_MAX);
          return ESP_ERR_INVALID_ARG;
76
      \ensuremath{//} Set the new PWM frequency of the given channel and timer configuration
78
      return ledc_set_freq(pwm_channel -> speed_mode, pwm_timer -> timer_num,
      frequency);
80 }
82 #endif // BUCK_CONVERTER_PWM
```

code/pwm.h

## **Application Code**

#### buck\_converter.h

```
1 #ifndef BUCK_CONVERTER
2 #define BUCK_CONVERTER
4 // Project includes
5 #include "pwm.h"
6 #include "adc.h"
7 #include "pid.h"
9 // Standard includes for math and print functions
10 #include <stdio.h>
11 #include <math.h>
12 #include <string.h>
_{14} // RTOS includes for error messages and task handling
#include "freertos/FreeRTOS.h"
16 #include "freertos/task.h"
17 #include "freertos/queue.h"
18 #include "esp_err.h"
20 void init_buck();
21 void control_loop(void *pvParameters);
23 /*
* PWM timer and channel structs
26 ledc_timer_config_t pwm_timer;
                                    // Create the PWM timer struct
27 ledc_channel_config_t pwm_channel; // Create the PWM channel struct
29 /*
30 * Constants for ADC conversion
32 static const float LOAD_R1 = 56237.0f; // Voltage divider values from buck output
33 static const float LOAD_R2 = 22035.0f + 995.2f;
35 static const float SUPPLY_R1 = 565300.0f; // Voltage divider values from buck
      output load
36 static const float SUPPLY_R2 = 119700.0f;
38 // Internal esp ADC structs
39 esp_adc v_supply;
* Constants, functions, and variables for PID controller
44 static const float VO = 3.0f; // The desired initial output of the converter.
46 // Controller gains
47 static const float KP = 0.16f;
48 static const float KI = 17.1f;
49 static const float KD = 0.0f;
```

```
51 // Controller Sample time period in ms
52 static const float TS = 2.0f;
53
54 #endif // BUCK_CONVERTER
```

code/buck\_converter.h

#### buck\_converter.c

```
1 // Include single header library function implementations
2 #define BUCK_CONVERTER_PWM_IMPL
3 #define BUCK_CONVERTER_ADC_IMPL
4 #define BUCK_CONVERTER_PID_IMPL
6 #include "buck_converter.h"
8 void init_buck()
9 {
      // PWM set up
      PWM_setup(&pwm_timer, &pwm_channel, 50000, 0);
11
12 }
13
14 /*
15
  * Output voltage PID control loop.
  * This function initialises the PID controller, and the internal ADC on the ESP32.
  * It will then enter the main control loop, and will compute the following:
17
18
          - Read the current buck converter load voltage. This is Measured using the
19
20
            ESP32 ADC, through the voltage divider defined by LOAD_R1 and LOAD_R2.
21
          - Calculate the equivalent duty cycle that would be used to produce the
22
            current measured load voltage.
23
          - Pass the target duty cycle, and current duty cycle through the PID
25
      controller
          - Update the buck converter duty cycle with the new controller output
27
28
          - Delay the task until the selected sample period defined by TS has elapsed
29
30
32 void control_loop(void *pvParameters)
33 ₹
      // Setup an input queue for receiving new target voltages
35
      QueueHandle_t target_voltage_queue = *(QueueHandle_t *)pvParameters;
36
      // Load voltage ADC set up and local variables
37
      uint16_t supply_voltage_buffer[5] = {0}; // Supply voltage rolling average
      buffer
39
      // Variables for storage and conversion of load voltage
      uint16_t load_adc_raw;
      float load_adc_voltage;
42
      float load_voltage;
43
      float measurment_duty;
44
      // Variables for storage and conversion of
      uint16_t supply_adc_raw;
47
      float supply_adc_voltage;
      float supply_voltage;
50
      // Initialise the ADC structs
51
      esp_adc v_load = {// Create adc struct for reading the load voltage
```

```
// adc input channel 0 (GPIO 36)
53
                          .adc_channel = 0,
54
                          // do not allow for a rolling average
56
                          .span = 0,
57
                          .buffer = NULL);
       esp_adc v_supply = {// Create adc struct for reading the supply voltage
60
                            // adc input channel 6 (GPIO 34)
61
                            .adc_channel = 6,
63
                            // 5 wide rolling average
64
                            .span = 5,
65
                            .buffer = (uint16_t *)&supply_voltage_buffer};
67
       // Init the load voltage adc and check to see if it was successful
68
       if (adc_init(&v_load) != ESP_OK)
69
           printf("Load voltage ADC init error!\n"); // If set up fails print the error
71
           vTaskDelete(NULL);
72
       }
73
74
       // Init the supply voltage adc and check to see if it was successful
75
       if (adc_init(&v_supply) != ESP_OK)
78
           printf("Supply voltage ADC init error!\n"); // If set up fails print the
       error
           vTaskDelete(NULL);
79
       }
80
81
       //PID set up
82
       float target_voltage = VO;
83
       float target_duty;
       // Create the PID struct
86
       PIDController voltage_controller = {
87
           // Controller gains
           .Kp = KP,
89
           .Ki = KI,
90
           .Kd = KD,
           // Differentiator low pass filter corner frequency
93
           .tau = 0.0f,
94
95
           // Min and max output duty cycles
           .limMin = 0.0f, // 0% duty cycle
97
           .limMax = 0.98f, // 95% duty cycle
           // Integrator wind-up min and max
100
           .limMinInt = -5.0f,
101
           .limMaxInt = 5.0f,
102
103
           // Sample time period
           .T = TS / 100.0f,
105
       };
106
```

107

```
// initialise the PID controller
108
      PID_controller_init(&voltage_controller);
109
       // Task variables
      TickType_t xLastWakeTime; // Hold the time stamp of the last wake time
112
113
      // Enter the PID control loop
114
      while (true)
       ₹
116
           xLastWakeTime = xTaskGetTickCount(); // Store the current tick count
118
119
           // Read the supply voltage ADC
           supply_adc_raw = adc_read(&v_supply);
           supply_adc_raw = rolling_average(&v_supply, supply_adc_raw);
122
           supply_adc_voltage = adc_conversion(supply_adc_raw);
           supply_voltage = (supply_adc_voltage * (SUPPLY_R1 + SUPPLY_R2)) / SUPPLY_R2;
           // Read the load voltage ADC
126
           load_adc_raw = adc_read(&v_load); // Take an adc reading
127
           load_adc_voltage = adc_conversion(load_adc_raw); // Convert this value to
      the voltage at the adc
          load_voltage = (load_adc_voltage * (LOAD_R1 + LOAD_R2)) / LOAD_R2; //
129
      Convert to the voltage at the buck converter load
130
           // Calculate the measured duty cycle and target duty cycle
           measurment_duty = load_voltage / supply_voltage; // Calculate the duty
      theoretical duty cycle of the output
           target_duty = target_voltage / supply_voltage; // Calculate the target duty
      cycle for the controller
134
           // Check for a new target voltage
           if (uxQueueMessagesWaiting(target_voltage_queue))
           { // If there is a new target voltage read it
               if (xQueueReceive(target_voltage_queue, &target_voltage, (TickType_t)1))
138
139
               {
                   target_duty = target_voltage / supply_voltage;
               }
141
           }
142
           // Update the duty cycle with the PID controller
           PID_controller_update(&voltage_controller, target_duty, measurment_duty);
145
146
           // Output the new duty cycle
147
           PWM_set_duty(&pwm_channel, voltage_controller.out);
148
149
           // Delay task until the provided time period is reached
150
           vTaskDelayUntil(&xLastWakeTime, TS);
151
       }
152
153 }
```

code/buck\_coverter.c

#### main.c

```
1 #include "buck_converter.h"
3 void app_main()
4 {
      init_buck();
6
      // Declare task handles
      TaskHandle_t VO_controller = NULL;
10
      QueueHandle_t target_voltage_queue;
      // initialise the queues
13
      target_voltage_queue = xQueueCreate(5, sizeof(float));
14
15
      // Create the output voltage control task
      xTaskCreatePinnedToCore(control_loop,
                                                                // Voltage control loop
17
      function
                               "Vout_controller",
                                                                // Task name
                                                                // Task stack size
                               2048,
                                (void *)&target_voltage_queue, // Function parameters
20
                                                                // Priority of the task
21
      (app_main has priority 1)
                                                                // Task handle
22
                               &VO_controller,
                               tskNO_AFFINITY);
                                                                // Core the task has
23
      been pinned to (No core selected)
24
      float input_voltage;
      char input_buffer[20] = {0};
26
      uint8_t buf_index = 0;
27
      while (true)
30
          char input = fgetc(stdin);
31
          if (input != 0xFF)
33
34
               input_buffer[buf_index++] = input; // read in the character, and
35
      increment the index
                                                   //echo the character back
              fputc(input, stdout);
36
37
               if ((input_buffer[buf_index - 1] == '\n'))
               {
40
                   input_voltage = strtof(input_buffer, NULL);
41
42
43
                   xQueueSend(target_voltage_queue, (void *)&input_voltage,
      (TickType_t)1);
44
                   printf("\nInput Target Voltage: %f\n", strtof(input_buffer, NULL));
45
                   buf_index = 0;
              }
47
          }
48
```

```
vTaskDelay(1);
50
51
           // printf("task stack unused: %d\n\n",
uxTaskGetStackHighWaterMark(VO_controller));
   // vTaskDelay(100);
52
53
           }
54
55 }
```

code/main.c